home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
SGI Developer Toolbox 6.1
/
SGI Developer Toolbox 6.1 - Disc 4.iso
/
public
/
ghostscript
/
src
/
zcspace2.c
< prev
next >
Wrap
C/C++ Source or Header
|
1994-08-01
|
9KB
|
284 lines
/* Copyright (C) 1992 Aladdin Enterprises. All rights reserved.
This file is part of Ghostscript.
Ghostscript is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY. No author or distributor accepts responsibility
to anyone for the consequences of using it or for whether it serves any
particular purpose or works at all, unless he says so in writing. Refer
to the Ghostscript General Public License for full details.
Everyone is granted permission to copy, modify and redistribute
Ghostscript, but only under the conditions described in the Ghostscript
General Public License. A copy of this license is supposed to have been
given to you along with Ghostscript so you can know your rights and
responsibilities. It should be in a file named COPYING. Among other
things, the copyright notice and this notice must be preserved on all
copies. */
/* zcspace2.c */
/* Level 2 color space operators for Ghostscript */
#include "ghost.h"
#include "errors.h"
#include "oper.h"
#include "gscolor.h"
#include "gxcolor.h"
#include "gscspace.h"
#include "gscolor2.h"
#include "dict.h"
#include "dparam.h"
#include "estack.h"
#include "iname.h" /* for name_eq */
#include "state.h"
#include "store.h"
/* Imported from gscolor2.c */
extern const gs_color_space_type
gs_color_space_type_Separation,
gs_color_space_type_Indexed,
gs_color_space_type_Pattern;
/* Imported from zcie.c */
extern int
zcolorspace_CIEBasedABC(P3(const ref *, gs_color_space *, ref_cie_procs *)),
zcolorspace_CIEBasedA(P3(const ref *, gs_color_space *, ref_cie_procs *));
/* Forward references */
typedef enum { cs_allow_base, cs_allow_paint, cs_allow_all } cs_allowed;
private int cspace_param(P4(const ref *, gs_color_space *, ref_color_procs *, cs_allowed));
/* Names of color spaces: */
static ref color_space_names[8];
#define name_DeviceGray color_space_names[(int)gs_color_space_index_DeviceGray]
#define name_DeviceRGB color_space_names[(int)gs_color_space_index_DeviceRGB]
#define name_DeviceCMYK color_space_names[(int)gs_color_space_index_DeviceCMYK]
#define name_CIEBasedABC color_space_names[(int)gs_color_space_index_CIEBasedABC]
#define name_CIEBasedA color_space_names[(int)gs_color_space_index_CIEBasedA]
#define name_Separation color_space_names[(int)gs_color_space_index_Separation]
#define name_Indexed color_space_names[(int)gs_color_space_index_Indexed]
#define name_Pattern color_space_names[(int)gs_color_space_index_Pattern]
/* Initialization */
private void
zcspace2_init(void)
{ static const names_def csn[] = {
/* Create the names of the color spaces. */
{ "DeviceGray", &name_DeviceGray },
{ "DeviceRGB", &name_DeviceRGB },
{ "DeviceCMYK", &name_DeviceCMYK },
{ "CIEBasedABC", &name_CIEBasedABC },
{ "CIEBasedA", &name_CIEBasedA },
{ "Separation", &name_Separation },
{ "Indexed", &name_Indexed },
{ "Pattern", &name_Pattern },
/* Mark the end of the initialized name list. */
names_def_end
};
init_names(csn);
}
/* currentcolorspace */
int
zcurrentcolorspace(register os_ptr op)
{ push(1);
if ( r_has_type(&istate.colorspace.array, t_null) )
{ /* Create the 1-element array on the fly. */
const gs_color_space *pcs = gs_currentcolorspace(igs);
make_tasv(op, t_array, a_readonly, 1, refs,
&color_space_names[(int)pcs->type->index]);
}
else
*op = istate.colorspace.array;
return 0;
}
/* setcolorspace */
int
zsetcolorspace(register os_ptr op)
{ gs_color_space cs;
ref_color_procs procs;
ref_colorspace cspace_old;
es_ptr ep = esp;
int code;
procs = istate.colorspace.procs;
code = cspace_param((const ref *)op, &cs, &procs, cs_allow_all);
if ( code < 0 )
{ esp = ep;
return code;
}
/* The color space installation procedure may refer to */
/* istate.colorspace.procs. */
cspace_old = istate.colorspace;
if ( r_has_type(op, t_name) )
make_null(&istate.colorspace.array); /* no params */
else
istate.colorspace.array = *op;
istate.colorspace.procs = procs;
code = gs_setcolorspace(igs, &cs);
if ( code < 0 )
{ istate.colorspace = cspace_old;
esp = ep;
return code;
}
pop(1);
return (esp == ep ? 0 : o_push_estack);
}
/* ------ Initialization procedure ------ */
op_def zcspace2_op_defs[] = {
{"0currentcolorspace", zcurrentcolorspace},
{"1setcolorspace", zsetcolorspace},
op_def_end(zcspace2_init)
};
/* ------ Internal procedures ------ */
/* Dummy indexed lookup procedure. */
private int
null_indexed_lookup_proc(int index, float *values)
{ return 0;
}
/* Extract the parameters for a color space. */
private int
cspace_param(const ref *pcsref, gs_color_space *pcs,
ref_color_procs *pcprocs, cs_allowed allow)
{ const ref *pcsa, *pcsn;
uint asize;
int csi;
int code;
if ( r_has_type(pcsref, t_array) )
{ check_read(*pcsref);
pcsa = pcsref->value.const_refs;
asize = r_size(pcsref);
if ( asize == 0 )
return_error(e_rangecheck);
}
else
{ pcsa = pcsref;
asize = 1;
}
pcsn = pcsa++;
asize--;
check_type(*pcsn, t_name);
for ( csi = 0; !name_eq(pcsn, &color_space_names[csi]); )
{ if ( ++csi == countof(color_space_names) )
return_error(e_rangecheck);
}
/* Note: to avoid having to undo allocations, we should make all */
/* checks before any recursive calls of cspace_params. */
/* Unfortunately, we can't do this in the case of Indexed spaces. */
switch ( (gs_color_space_index)csi )
{
case gs_color_space_index_DeviceGray:
if ( asize != 0 )
return_error(e_rangecheck);
pcs->type = &gs_color_space_type_DeviceGray;
break;
case gs_color_space_index_DeviceRGB:
if ( asize != 0 )
return_error(e_rangecheck);
pcs->type = &gs_color_space_type_DeviceRGB;
break;
case gs_color_space_index_DeviceCMYK:
if ( asize != 0 )
return_error(e_rangecheck);
pcs->type = &gs_color_space_type_DeviceCMYK;
break;
case gs_color_space_index_CIEBasedABC:
if ( asize != 1 )
return_error(e_rangecheck);
code = zcolorspace_CIEBasedABC(pcsa, pcs, &pcprocs->cie);
if ( code < 0 )
return code;
/*pcs->type = &gs_color_space_type_CIEBasedABC;*/ /* set by zcolorspace... */
break;
case gs_color_space_index_CIEBasedA:
if ( asize != 1 )
return_error(e_rangecheck);
code = zcolorspace_CIEBasedA(pcsa, pcs, &pcprocs->cie);
if ( code < 0 )
return code;
/*pcs->type = &gs_color_space_type_CIEBasedA;*/ /* set by zcolorspace... */
break;
case gs_color_space_index_Separation:
if ( allow == cs_allow_base )
return_error(e_rangecheck);
if ( asize != 3 )
return_error(e_rangecheck);
check_type(*pcsa, t_name);
check_proc(pcsa[2]);
code = cspace_param(pcsa + 1, (gs_color_space *)&pcs->params.separation.alt_space, pcprocs, cs_allow_base);
if ( code < 0 )
return code;
pcprocs->special.separation.layer_name = pcsa[0];
pcprocs->special.separation.tint_transform = pcsa[2];
pcs->type = &gs_color_space_type_Separation;
break;
case gs_color_space_index_Indexed:
if ( allow == cs_allow_base )
return_error(e_rangecheck);
if ( asize != 3 )
return_error(e_rangecheck);
check_type(pcsa[1], t_integer);
if ( pcsa[1].value.intval < 0 || pcsa[1].value.intval > 4095 )
return_error(e_rangecheck);
pcs->params.indexed.hival = pcsa[1].value.intval;
/* We must get the base space now, rather than later, */
/* so we can check the length of the table against */
/* the num_components of the base space. */
code = cspace_param(pcsa, (gs_color_space *)&pcs->params.indexed.base_space, pcprocs, cs_allow_base);
if ( code < 0 )
return code;
if ( r_has_type(&pcsa[2], t_string) )
{ check_read(pcsa[2]);
if ( r_size(&pcsa[2]) !=
(pcs->params.indexed.hival + 1) *
pcs->params.indexed.base_space.type->num_components
)
return_error(e_rangecheck);
pcs->params.indexed.lookup.table =
pcsa[2].value.const_bytes;
pcs->params.indexed.use_proc = 0;
make_null(&pcprocs->special.index_proc);
}
else
{ check_proc(pcsa[2]);
pcs->params.indexed.lookup.proc =
null_indexed_lookup_proc;
pcs->params.indexed.use_proc = 1;
pcprocs->special.index_proc = pcsa[2];
}
pcs->type = &gs_color_space_type_Indexed;
break;
case gs_color_space_index_Pattern:
if ( allow != cs_allow_all )
return_error(e_rangecheck);
switch ( asize )
{
case 0: /* no base space */
pcs->params.pattern.has_base_space = 0;
break;
default:
return_error(e_rangecheck);
case 1:
pcs->params.pattern.has_base_space = 1;
code = cspace_param(pcsa,
(gs_color_space *)&pcs->params.pattern.base_space,
pcprocs, cs_allow_paint);
if ( code < 0 )
return code;
}
pcs->type = &gs_color_space_type_Pattern;
break;
default:
return_error(e_typecheck);
}
return 0;
}